import { getBillingType } from "@/internal/products/prices/priceUtils.js";
import RecaseError from "@/utils/errorUtils.js";
import { generateId } from "@/utils/genUtils.js";
import {
  AllowanceType,
  BillingInterval,
  BillingType,
  Entitlement,
  ErrCode,
  FixedPriceConfig,
  FixedPriceConfigSchema,
  Price,
  PriceType,
  UsagePriceConfig,
  UsagePriceConfigSchema,
} from "@autumn/shared";

export const constructPrice = ({
  name,
  config,
  orgId,
  internalProductId,
  isCustom = false,
}: {
  name: string;
  config: UsagePriceConfig | FixedPriceConfig;
  orgId: string;
  internalProductId: string;
  isCustom: boolean;
}) => {
  return {
    id: generateId("pr"),
    org_id: orgId,
    internal_product_id: internalProductId,
    created_at: Date.now(),
    billing_type: getBillingType(config),
    is_custom: isCustom,

    name,
    config,
  };
};

// GET PRICES
const validatePrice = (
  price: Price,
  relatedEnt?: Entitlement | undefined | null,
) => {
  if (!price.config?.type) {
    throw new RecaseError({
      message: "Missing `type` field in price config",
      code: ErrCode.InvalidPriceConfig,
      statusCode: 400,
    });
  }

  if (price.config?.type == PriceType.Fixed) {
    FixedPriceConfigSchema.parse(price.config);
  } else {
    UsagePriceConfigSchema.parse(price.config);

    const config = price.config! as UsagePriceConfig;

    if (config.usage_tiers.length == 0) {
      throw new RecaseError({
        message: "Usage based prices should have at least one tier",
        code: ErrCode.InvalidPriceConfig,
        statusCode: 400,
      });
    }

    if (relatedEnt?.allowance_type == AllowanceType.Unlimited) {
      if (config.interval == BillingInterval.OneOff) {
        throw new RecaseError({
          message: `Usage-based price cannot have unlimited allowance (${relatedEnt.feature_id})`,
          code: ErrCode.InvalidPriceConfig,
          statusCode: 400,
        });
      }
    }

    const billingType = getBillingType(config);
    if (billingType == BillingType.UsageInArrear) {
      if (config.interval == BillingInterval.OneOff) {
        throw new RecaseError({
          message: "One off prices must be billed at start of period",
          code: ErrCode.InvalidPriceConfig,
          statusCode: 400,
        });
      }
    }
  }

  return {
    valid: true,
    error: null,
  };
};

export const tiersAreSame = (tiers1: any[], tiers2: any[]) => {
  if (tiers1.length !== tiers2.length) return false;
  for (let i = 0; i < tiers1.length; i++) {
    let tier1 = tiers1[i];
    let tier2 = tiers2[i];
    // if (tier1.to !== tier2.to) return false;

    // Only compare to if not last tier
    if (i !== tiers1.length - 1) {
      if (tier1.to !== tier2.to) return false;
    }

    if (tier1.amount !== tier2.amount) return false;
  }
  return true;
};

export const pricesAreSame = (
  price1: Price,
  price2: Price,
  logDifferences = false,
) => {
  // if (price1.name !== price2.name) return false;

  const config1 = price1.config!;
  const config2 = price2.config!;
  // if (config1.type !== config2.type) return false; // shouldn't be able to change, but just in case...

  if (config1.type === PriceType.Fixed) {
    const fixedConfig1 = FixedPriceConfigSchema.parse(config1);
    const fixedConfig2 = FixedPriceConfigSchema.parse(config2);

    // 1. Check amount same
    let diffs = {
      amount: {
        condition: fixedConfig1.amount !== fixedConfig2.amount,
        message: `Amount different: ${fixedConfig1.amount} !== ${fixedConfig2.amount}`,
      },
      interval: {
        condition: fixedConfig1.interval !== fixedConfig2.interval,
        message: `Interval different: ${fixedConfig1.interval} !== ${fixedConfig2.interval}`,
      },
    };

    let pricesAreDiff = Object.values(diffs).some((d) => d.condition);

    if (pricesAreDiff) {
      console.log("Fixed price different");
      console.log(
        "Differences:",
        Object.values(diffs)
          .filter((d) => d.condition)
          .map((d) => d.message),
      );
    }

    return !pricesAreDiff;
  } else {
    const usageConfig1 = UsagePriceConfigSchema.parse(config1);
    const usageConfig2 = UsagePriceConfigSchema.parse(config2);

    let diffs = {
      should_prorate: {
        condition: usageConfig1.should_prorate !== usageConfig2.should_prorate,
        message: `Should prorate different: ${usageConfig1.should_prorate} !== ${usageConfig2.should_prorate}`,
      },
      bill_when: {
        condition: usageConfig1.bill_when !== usageConfig2.bill_when,
        message: `Bill when different: ${usageConfig1.bill_when} !== ${usageConfig2.bill_when}`,
      },
      billing_units: {
        condition: usageConfig1.billing_units !== usageConfig2.billing_units,
        message: `Billing units different: ${usageConfig1.billing_units} !== ${usageConfig2.billing_units}`,
      },
      interval: {
        condition: usageConfig1.interval !== usageConfig2.interval,
        message: `Interval different: ${usageConfig1.interval} !== ${usageConfig2.interval}`,
      },
      internal_feature_id: {
        condition:
          usageConfig1.internal_feature_id !== usageConfig2.internal_feature_id,
        message: `Internal feature ID different: ${usageConfig1.internal_feature_id} !== ${usageConfig2.internal_feature_id}`,
      },
      feature_id: {
        condition: usageConfig1.feature_id !== usageConfig2.feature_id,
        message: `Feature ID different: ${usageConfig1.feature_id} !== ${usageConfig2.feature_id}`,
      },
      usage_tiers: {
        condition: !tiersAreSame(
          usageConfig1.usage_tiers,
          usageConfig2.usage_tiers,
        ),
        message: `Usage tiers different: ${usageConfig1.usage_tiers.map(
          (t) => `${t.to} (${t.amount})`,
        )} !== ${usageConfig2.usage_tiers.map((t) => `${t.to} (${t.amount})`)}`,
      },
      // stripe_price_id: {
      //   condition:
      //     usageConfig1.stripe_price_id !== usageConfig2.stripe_price_id,
      //   message: `Stripe price ID different: ${usageConfig1.stripe_price_id} !== ${usageConfig2.stripe_price_id}`,
      // },
      // stripe_placeholder_price_id: {
      //   condition:
      //     usageConfig1.stripe_placeholder_price_id !==
      //     usageConfig2.stripe_placeholder_price_id,
      //   message: `Stripe placeholder price ID different: ${usageConfig1.stripe_placeholder_price_id} !== ${usageConfig2.stripe_placeholder_price_id}`,
      // },
      // stripe_meter_id: {
      //   condition:
      //     usageConfig1.stripe_meter_id !== usageConfig2.stripe_meter_id,
      //   message: `Stripe meter ID different: ${usageConfig1.stripe_meter_id} !== ${usageConfig2.stripe_meter_id}`,
      // },
      // stripe_product_id: {
      //   condition:
      //     usageConfig1.stripe_product_id !== usageConfig2.stripe_product_id,
      //   message: `Stripe product ID different: ${usageConfig1.stripe_product_id} !== ${usageConfig2.stripe_product_id}`,
      // },
    };

    let pricesAreDiff = Object.values(diffs).some((d) => d.condition);

    if (pricesAreDiff && logDifferences) {
      console.log(`Usage price different: ${usageConfig1.feature_id}`);
      console.log(
        "Differences:",
        Object.values(diffs)
          .filter((d) => d.condition)
          .map((d) => d.message),
      );
    }

    return !pricesAreDiff;
  }
};
